1   /**
2    * Copyright (C) 2006-2019 INRIA and contributors
3    *
4    * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
5    */
6   package spoon.reflect.visitor;
7   
8   import spoon.reflect.code.CtExpression;
9   import spoon.reflect.code.CtFieldAccess;
10  import spoon.reflect.code.CtInvocation;
11  import spoon.reflect.code.CtTargetedExpression;
12  import spoon.reflect.code.CtTypeAccess;
13  import spoon.reflect.declaration.CtElement;
14  import spoon.reflect.declaration.CtExecutable;
15  import spoon.reflect.declaration.CtField;
16  import spoon.reflect.declaration.CtMethod;
17  import spoon.reflect.declaration.CtNamedElement;
18  import spoon.reflect.declaration.CtType;
19  import spoon.reflect.declaration.CtTypeMember;
20  import spoon.reflect.path.CtRole;
21  import spoon.reflect.reference.CtExecutableReference;
22  import spoon.reflect.reference.CtFieldReference;
23  import spoon.reflect.reference.CtPackageReference;
24  import spoon.reflect.reference.CtTypeReference;
25  import spoon.support.Experimental;
26  
27  
28  /**
29   * Detects conflicts needed to be required be a fully-qualified name.
30   *
31   * 1) Example: conflict of field name with an variable name and fixes it by making field target explicit.
32   * <code><pre>
33   * class A {
34   *  int xxx;
35   *  void m(String xxx) {
36   *    this.xxx //the target `this.` must be explicit, otherwise parameter `String xxx` hides it
37   *  }
38   * }
39   *</pre></code>
40   *
41   * 2) Example: conflict of package name with an variable name and fixes it by making field target implicit.
42   * <code><pre>
43   * class A {
44   *  int com;
45   *  void m() {
46   *    com.package.Type.doSomething(); //the package `com` is in conflict with field `com`. Must be imported
47   *  }
48   * }
49   *</pre></code>
50   * and fixes them by call of {@link CtElement#setImplicit(boolean)} and {@link CtTypeReference#setSimplyQualified(boolean)}
51   */
52  @Experimental
53  public class ImportConflictDetector extends ImportAnalyzer<LexicalScope> {
54  
55  	@Override
56  	protected LexicalScopeScanner createScanner() {
57  		return new LexicalScopeScanner();
58  	}
59  
60  	@Override
61  	protected LexicalScope getScannerContextInformation() {
62  		return ((LexicalScopeScanner) scanner).getCurrentLexicalScope();
63  	}
64  
65  	@Override
66  	protected void handleTargetedExpression(CtTargetedExpression<?, ?> targetedExpression, LexicalScope nameScope) {
67  		CtExpression<?> target = targetedExpression.getTarget();
68  		if (target == null) {
69  			return;
70  		}
71  		if (targetedExpression instanceof CtFieldAccess<?>) {
72  			CtFieldAccess<?> fieldAccess = (CtFieldAccess<?>) targetedExpression;
73  			if (target.isImplicit()) {
74  				/*
75  				 * target is implicit, check whether there is no conflict with an local variable, catch variable or parameter
76  				 * in case of conflict make it explicit, otherwise the field access is shadowed by that variable.
77  				 * Search for potential variable declaration until we found a class which declares or inherits this field
78  				 */
79  				final CtField<?> field = fieldAccess.getVariable().getFieldDeclaration();
80  				if (field != null) {
81  					final String fieldName = field.getSimpleName();
82  					nameScope.forEachElementByName(fieldName, named -> {
83  						if (named instanceof CtMethod) {
84  							//the methods with same name are no problem for field access
85  							return null;
86  						}
87  						if (named == field) {
88  							return true;
89  						}
90  						//another variable declaration was found which is hiding the field declaration for this field access. Make the field access explicit
91  						target.setImplicit(false);
92  						return false;
93  					});
94  				}
95  			}
96  			if (!target.isImplicit()) {
97  				//the target should be visible in sources
98  				if (target instanceof CtTypeAccess) {
99  					//the type has to be visible in sources
100 					CtTypeAccess<?> typeAccess = (CtTypeAccess<?>) target;
101 					CtTypeReference<?> accessedTypeRef = typeAccess.getAccessedType();
102 					checkConflictOfTypeReference(nameScope, accessedTypeRef);
103 				}
104 			}
105 		}
106 	}
107 
108 	@Override
109 	protected void handleTypeReference(CtTypeReference<?> ref, LexicalScope nameScope, CtRole role) {
110 		if (ref.isImplicit()) {
111 			/*
112 			 * the reference is implicit. E.g. `assertTrue();`
113 			 * when the type `org.junit.Assert` is implicit
114 			 */
115 			//check if targeted expression is in conflict
116 			CtTargetedExpression<?, ?> targetedExpr = getParentIfType(getParentIfType(ref, CtTypeAccess.class), CtTargetedExpression.class);
117 			if (targetedExpr instanceof CtInvocation<?>) {
118 				CtInvocation<?> invocation = (CtInvocation<?>) targetedExpr;
119 				CtExecutableReference<?> importedReference = invocation.getExecutable();
120 				CtExecutable<?> importedElement = importedReference.getExecutableDeclaration();
121 				if (importedElement == null) {
122 					//we have no access to executable - probably no class path mode
123 					//What to do? Keep it as it is? Or make it fully qualified?
124 					//keep it as it is for now.
125 					return;
126 				}
127 				if (importedElement instanceof CtMethod) {
128 					//check if statically imported field or method simple name is not in conflict
129 					nameScope.forEachElementByName(importedReference.getSimpleName(), named -> {
130 						if (named instanceof CtMethod<?>) {
131 							//the method call can be in conflict with Method name only
132 							if (isSameStaticImport(named, importedElement)) {
133 								//we have found import of the same method
134 								return true;
135 							}
136 							ref.setImplicit(false);
137 							ref.setSimplyQualified(true);
138 							return false;
139 						}
140 						//no conflict with type or field name
141 						return null;
142 					});
143 				}
144 			} else if (targetedExpr instanceof CtFieldAccess<?>) {
145 				CtFieldAccess<?> fieldAccess = (CtFieldAccess<?>) targetedExpr;
146 				CtFieldReference<?> importedReference = fieldAccess.getVariable();
147 				CtElement importedElement = importedReference.getFieldDeclaration();
148 				if (importedElement == null) {
149 					//we have no access to executable - probably no class path mode
150 					//What to do? Keep it as it is? Or make it fully qualified?
151 					//keep it as it is for now.
152 					return;
153 				}
154 				//check if statically imported field or method simple name is not in conflict
155 				nameScope.forEachElementByName(importedReference.getSimpleName(), named -> {
156 					if (named instanceof CtMethod<?>) {
157 						//field access cannot be in conflict with method name
158 						return null;
159 					}
160 					if (named == importedElement) {
161 						//we have found import of the same field
162 						return true;
163 					}
164 					//else there is a conflict. Make type explicit and package implicit
165 					ref.setImplicit(false);
166 					ref.setSimplyQualified(true);
167 					return false;
168 				});
169 			}
170 			//else do nothing like in case of implicit type of lambda parameter
171 			//`(e) -> {...}`
172 		}
173 		if (!ref.isImplicit() && ref.isSimplyQualified()) {
174 			/*
175 			 * the package is implicit. E.g. `Assert.assertTrue`
176 			 * where package `org.junit` is implicit
177 			 */
178 			String refQName = ref.getQualifiedName();
179 			//check if type simple name is not in conflict
180 			nameScope.forEachElementByName(ref.getSimpleName(), named -> {
181 				if (named instanceof CtMethod) {
182 					//the methods with same name are no problem for field access
183 					return null;
184 				}
185 				if (named instanceof CtType) {
186 					CtType<?> type = (CtType<?>) named;
187 					if (refQName.equals(type.getQualifiedName())) {
188 						//we have found a declaration of type of the ref. We can use implicit
189 						return true;
190 					}
191 				}
192 				//we have found a variable, field or type of different name
193 				ref.setImplicit(false);
194 				ref.setSimplyQualified(false);
195 				return false;
196 			});
197 		} //else it is already fully qualified
198 		checkConflictOfTypeReference(nameScope, ref);
199 	}
200 
201 	/**
202 	 * if typeRef package name or simple name is in conflict with any name from nameScope then
203 	 * solve that conflict by package or type implicit
204 	 */
205 	private void checkConflictOfTypeReference(LexicalScope nameScope, CtTypeReference<?> typeRef) {
206 		if (typeRef == null) {
207 			return;
208 		}
209 		if (!typeRef.isSimplyQualified()) {
210 			//we have to print fully qualified type name
211 			//is the first part of package name in conflict with something else?
212 			if (isPackageNameConflict(nameScope, typeRef)) {
213 				//the package must be imported, then simple name might be used in this scope
214 				typeRef.setSimplyQualified(true);
215 				if (isSimpleNameConflict(nameScope, typeRef)) {
216 					//there is conflict with simple name too
217 					typeRef.setImplicit(true);
218 				}
219 			}
220 		} else {
221 			if (!typeRef.isImplicit()) {
222 				if (isSimpleNameConflict(nameScope, typeRef)) {
223 					//there is conflict with simple name
224 					//use qualified name
225 					typeRef.setSimplyQualified(false);
226 				}
227 			}
228 		}
229 	}
230 
231 	private boolean isPackageNameConflict(LexicalScope nameScope, CtTypeReference<?> typeRef) {
232 		String fistPackageName = getFirstPackageQName(typeRef);
233 		if (fistPackageName != null) {
234 			return Boolean.TRUE == nameScope.forEachElementByName(fistPackageName, named -> {
235 				if (named instanceof CtMethod) {
236 					//same method name is not a problem
237 					return null;
238 				}
239 				//the package name is in conflict with field name, variable or type name
240 				return Boolean.TRUE;
241 			});
242 		}
243 		return false;
244 	}
245 
246 	private boolean isSimpleNameConflict(LexicalScope nameScope, CtTypeReference<?> typeRef) {
247 		String typeQName = typeRef.getQualifiedName();
248 		//the package name is in conflict check whether type name is in conflict
249 		return Boolean.TRUE == nameScope.forEachElementByName(typeRef.getSimpleName(), named -> {
250 			if (named instanceof CtMethod) {
251 				//same method name is not a problem
252 				return null;
253 			}
254 			if (named instanceof CtType) {
255 				CtType<?> type = (CtType<?>) named;
256 				if (typeQName.equals(type.getQualifiedName())) {
257 					//we found the referenced type declaration -> ok, we can use simple name in this scope
258 					return Boolean.FALSE;
259 				}
260 				//we found another type with same simple name -> conflict
261 			}
262 			//there is conflict with simple name
263 			return Boolean.TRUE;
264 		});
265 	}
266 
267 	private String getFirstPackageQName(CtTypeReference<?> typeRef) {
268 		if (typeRef != null) {
269 			CtPackageReference packRef = typeRef.getPackage();
270 			if (packRef != null) {
271 				String qname = packRef.getQualifiedName();
272 				if (qname != null && qname.length() > 0) {
273 					int idx = qname.indexOf('.');
274 					if (idx < 0) {
275 						idx = qname.length();
276 					}
277 					return qname.substring(0, idx);
278 				}
279 			}
280 		}
281 		return null;
282 	}
283 	/**
284 	 * @return true if two methods can come from same static import
285 	 *
286 	 * For example `import static org.junit.Assert.assertEquals;`
287 	 * imports methods with signatures:
288 	 * assertEquals(Object, Object)
289 	 * assertEquals(Object[], Object[])
290 	 * assertEquals(long, long)
291 	 * ...
292 	 */
293 	private static boolean isSameStaticImport(CtNamedElement m1, CtNamedElement m2) {
294 		if (m1 instanceof CtTypeMember && m2 instanceof CtTypeMember) {
295 			if (m1.getSimpleName().equals(m2.getSimpleName())) {
296 				CtType<?> declType1 = ((CtTypeMember) m1).getDeclaringType();
297 				CtType<?> declType2 = ((CtTypeMember) m2).getDeclaringType();
298 				//may be we should check isSubTypeOf instead
299 				return declType1 == declType2;
300 			}
301 		}
302 		return false;
303 	}
304 }